package xyz.zhuht.algorithm.difficult;

/**
 * @author haitao zhu
 * @date 2020/7/5 20:44
 * 44.通配符匹配
 * 给定一个字符串 (s) 和一个字符模式 (p) ，实现一个支持 '?' 和 '*' 的通配符匹配。
 * <p>
 * '?' 可以匹配任何单个字符。
 * '*' 可以匹配任意字符串（包括空字符串）。
 * 两个字符串完全匹配才算匹配成功。
 * <p>
 * 说明:
 * <p>
 * s 可能为空，且只包含从 a-z 的小写字母。
 * p 可能为空，且只包含从 a-z 的小写字母，以及字符 ? 和 *。
 * 示例 1:
 * <p>
 * 输入:
 * s = "aa"
 * p = "a"
 * 输出: false
 * 解释: "a" 无法匹配 "aa" 整个字符串。
 * 示例 2:
 * <p>
 * 输入:
 * s = "aa"
 * p = "*"
 * 输出: true
 * 解释: '*' 可以匹配任意字符串。
 * 示例 3:
 * <p>
 * 输入:
 * s = "cb"
 * p = "?a"
 * 输出: false
 * 解释: '?' 可以匹配 'c', 但第二个 'a' 无法匹配 'b'。
 * 示例 4:
 * <p>
 * 输入:
 * s = "adceb"
 * p = "*a*b"
 * 输出: true
 * 解释: 第一个 '*' 可以匹配空字符串, 第二个 '*' 可以匹配字符串 "dce".
 * 示例 5:
 * <p>
 * 输入:
 * s = "acdcb"
 * p = "a*c?b"
 * 输出: false
 * <p>
 * 来源：力扣（LeetCode）
 * 链接：https://leetcode-cn.com/problems/wildcard-matching
 * 著作权归领扣网络所有。商业转载请联系官方授权，非商业转载请注明出处。
 */
public class TongPeiFuPiPei {

  public boolean isMatch(String s, String p) {
    int sRight = s.length(), pRight = p.length();
    while (sRight > 0 && pRight > 0 && p.charAt(pRight - 1) != '*') {
      if (charMatch(s.charAt(sRight - 1), p.charAt(pRight - 1))) {
        --sRight;
        --pRight;
      } else {
        return false;
      }
    }

    if (pRight == 0) {
      return sRight == 0;
    }

    int sIndex = 0, pIndex = 0;
    int sRecord = -1, pRecord = -1;

    while (sIndex < sRight && pIndex < pRight) {
      if (p.charAt(pIndex) == '*') {
        ++pIndex;
        sRecord = sIndex;
        pRecord = pIndex;
      } else if (charMatch(s.charAt(sIndex), p.charAt(pIndex))) {
        ++sIndex;
        ++pIndex;
      } else if (sRecord != -1 && sRecord + 1 < sRight) {
        ++sRecord;
        sIndex = sRecord;
        pIndex = pRecord;
      } else {
        return false;
      }
    }

    return allStars(p, pIndex, pRight);
  }

  public boolean allStars(String str, int left, int right) {
    for (int i = left; i < right; ++i) {
      if (str.charAt(i) != '*') {
        return false;
      }
    }
    return true;
  }

  public boolean charMatch(char u, char v) {
    return u == v || v == '?';
  }

}
